处理事件

  如前所述,要处理事件,需要提供一个事件处理方法来订阅事件,该方法的返回类型和参数应该匹配事件指定的委托。下面的示例使用一个简单的计时器对象引发事件,调用一个处理方法。

    using System;
    using Sytem.Collections.Generic;
    using System.Linq;
    using System.Text;
    using System.Threading.Tasks;
    using System.Timers;

    namespace Ch13Ex01
    {
        class Program
        {
            static int counter = 0;

            static string displayStrng = 
                "This string will appear one letter at a time.";

            static void Main(string[] args)
            {
                Timer myTimer = new Timer(100);
                myTimer.Elapsed += new ElapsedEventHandler(WriteChar);
                myTimer.Start();
                System.Threading.Thread.Sleep(200);
                Console.ReadKey();
            }

            static void WriteChar(object source, ElapsedEventArgs e)
            {
                Console.Write(displayString[counter++ % displayString.Length]);
            }
        }
    }

  示例的说明

  用于引发事件的对象是System.Timers.Timer类的一个实例。使用一个时间段(以毫秒为单位)来初始化该对象。当使用Start()方法启动Timer对象时,就引发一系列事件,根据指定的时间段来引发事件。Main()用100毫秒初始化Timer对象,所以在启动该对象后,1秒钟内将引发10次事件:

    static void Main(string[] args)
    {
        Timer myTimer = new Timer(100);

  Timer对象有一个Elapsed事件,这个事件要求事件处理程序必须匹配System.Timers.ElapsedEventHandler委托类型的返回类型和参数,该委托是.NET Framework中定义的标准委托之一,指定了如下所示的返回类型和参数:

    void <MethodName>(object source, ElapsedVenetArgs e);

  Timer对象的第一个参数是它本身的引用,第二个参数则是ElapsedEventArgs对象的一个实例。现在可以不考虑这些参数,后面将论述它们。

  在代码中,有一个匹配该返回类型和参数的方法:

    static void WriteChar(object source, ElapsedEventArgs e)
    {
        Console.Write(displayString[counter++ % displayString.Length]);
    }

  这个方法使用Program的两个静态字段counter和displayString来显示一个字符。每次调用该方法时,显示的字符都不相同。

  下一个任务是把这个处理程序与事件关联起来---即订阅它。为此,可以使用+=运算符,给事件添加一个处理程序,其形式是使用事件处理方法初始化的一个新委托实例:

    static void Main(string[] args)
    {
        Timer myTimer = new Timer(100);
        myTimer.Elapsed += new ElapsedEventHandler(WriteChar);
    }

  这个命令(使用有点古怪的语法,专用于委托)在列表中添加一个处理程序,当引发Elapsed事件时,就会调用该处理程序。可给这个列表添加任意多个处理程序,只要它们满足指定的条件即可。当引发事件时,会依次调用每个处理程序。

  Main()剩余的任务是启动计时器:

    myTimer.Start();

  我们不想在处理完任何事件前终止应用程序,所以要让Main()函数一直执行。最简单的方式是请求用户输入,因为这个命令要在用户按下任意键后,才会停止处理。

    Console.ReadKey();

  这里,Main()中的处理会停止,但Timer对象中的处理将继续。当该对象引发事件时,就调用WriteChar()方法,同时该方法运行Console.ReadLine()语句。使用System.Threading.Thread.Sleep(200)语句是为了让计时器有机会把消息发送给控制台应用程序。

  注意,可使用上一章介绍的方法组概念来稍简化添加事件处理程序的语法:

    myTimer.Elapsed += WriteChar;

  最终结果是完全相同的,但不必显式指定委托类型,编译器会根据使用事件的上下文来指定它。但是,许多程序员不喜欢这个语法,因为它降低了可读性---不再能一眼看出使用了什么委托类型。如果喜欢,就可以使用这个语法。但为了清晰起见,本章使用的所有委托都显式指定。

🔚